home *** CD-ROM | disk | FTP | other *** search
/ User's Choice Windows CD / User's Choice Windows CD (CMS Software)(1993).iso / utility1 / gs261src.zip / ZCHAR2.C < prev    next >
C/C++ Source or Header  |  1993-05-27  |  7KB  |  269 lines

  1. /* Copyright (C) 1992, 1993 Aladdin Enterprises.  All rights reserved.
  2.  
  3. This file is part of Ghostscript.
  4.  
  5. Ghostscript is distributed in the hope that it will be useful, but
  6. WITHOUT ANY WARRANTY.  No author or distributor accepts responsibility
  7. to anyone for the consequences of using it or for whether it serves any
  8. particular purpose or works at all, unless he says so in writing.  Refer
  9. to the Ghostscript General Public License for full details.
  10.  
  11. Everyone is granted permission to copy, modify and redistribute
  12. Ghostscript, but only under the conditions described in the Ghostscript
  13. General Public License.  A copy of this license is supposed to have been
  14. given to you along with Ghostscript so you can know your rights and
  15. responsibilities.  It should be in a file named COPYING.  Among other
  16. things, the copyright notice and this notice must be preserved on all
  17. copies.  */
  18.  
  19. /* zchar2.c */
  20. /* Level 2 character operators for Ghostscript */
  21. #include "ghost.h"
  22. #include "errors.h"
  23. #include "oper.h"
  24. #include "gschar.h"
  25. #include "gsmatrix.h"        /* for gxfont.h */
  26. #include "gxfixed.h"        /* for gxfont.h */
  27. #include "gxfont.h"
  28. #include "alloc.h"
  29. #include "estack.h"
  30. #include "font.h"
  31. #include "iname.h"
  32. #include "state.h"
  33. #include "store.h"
  34. #include "stream.h"
  35. #include "bnum.h"
  36.  
  37. /* Imported from zchar.c */
  38. extern int op_show_setup(P2(os_ptr, int /*bool*/));
  39. extern int op_show_continue(P1(os_ptr));
  40. extern int op_show_continue_dispatch(P2(os_ptr, int));
  41. extern gs_show_enum *op_show_find(P0());
  42. extern gs_show_enum *op_show_senum(P0());
  43. extern ref *op_show_psslot(P0());
  44. extern void op_show_free(P0());
  45.  
  46. /* Table of continuation procedures. */
  47. private int xshow_continue(P1(os_ptr));
  48. private int yshow_continue(P1(os_ptr));
  49. private int xyshow_continue(P1(os_ptr));
  50. static op_proc_p xyshow_continues[4] = {
  51.     0, xshow_continue, yshow_continue, xyshow_continue
  52. };
  53.  
  54. /* Forward references */
  55. private int cshow_continue(P1(os_ptr));
  56. private int moveshow(P2(os_ptr, int));
  57. private int moveshow_continue(P2(os_ptr, int));
  58.  
  59. /* <proc> <string> cshow - */
  60. private int
  61. zcshow(os_ptr op)
  62. {    int code;
  63.     check_proc(op[-1]);
  64.     if ( (code = op_show_setup(op, 1)) < 0 ) return code;
  65.     if ( (code = gs_cshow_n_init(op_show_senum(), igs, (char *)op->value.bytes, r_size(op))) < 0 )
  66.        {    op_show_free();
  67.         return code;
  68.        }
  69.     *op_show_psslot() = op[-1];        /* save kerning proc */
  70.     pop(2);  op -= 2;
  71.     return cshow_continue(op);
  72. }
  73. private int
  74. cshow_continue(os_ptr op)
  75. {    gs_show_enum *penum = op_show_senum();
  76.     int code = gs_show_next(penum);
  77.     if ( code != gs_show_move )
  78.     {    code = op_show_continue_dispatch(op, code);
  79.         if ( code == o_push_estack )    /* must be gs_show_render */
  80.         {    make_op_estack(esp - 1, cshow_continue);
  81.         }
  82.         else if ( code < 0 )
  83.             goto errx;
  84.         return code;
  85.     }
  86.     /* Push the character code and width, and call the procedure. */
  87.     {    gs_show_enum *penum = op_show_senum();
  88.         ref *pslot = op_show_psslot();
  89.         gs_point wpt;
  90.         gs_show_current_width(penum, &wpt);
  91.         push(3);
  92.         make_int(op - 2, gs_show_current_char(penum));
  93.         make_real(op - 1, wpt.x);
  94.         make_real(op, wpt.y);
  95.         push_op_estack(cshow_continue);
  96.         *++esp = *pslot;    /* user procedure */
  97.     }
  98.     return o_push_estack;
  99. errx:    op_show_free();
  100.     return code;
  101. }
  102.  
  103. /* <charname> glyphshow - */
  104. private int
  105. zglyphshow(os_ptr op)
  106. {    int code;
  107.     check_type(*op, t_name);
  108.     if ( (code = op_show_setup(op, 0)) < 0 )
  109.         return code;
  110.     if ( (code = gs_glyphshow_init(op_show_senum(), igs,
  111.         (gs_glyph)name_index(op))) < 0
  112.        )
  113.     {    op_show_free();
  114.         return code;
  115.     }
  116.     pop(1);  op--;
  117.     return op_show_continue(op);
  118. }
  119.  
  120. /* - rootfont <font> */
  121. private int
  122. zrootfont(os_ptr op)
  123. {    gs_font *pfont = gs_rootfont(op_show_find(), igs);
  124.     push(1);
  125.     *op = ((font_data *)(pfont->client_data))->dict;
  126.     return 0;
  127. }
  128.  
  129. /* <w0x> <w0y> <llx> <lly> <urx> <ury> <w1x> <w1y> <vx> <vy> setcachedevice2 - */
  130. private int
  131. zsetcachedevice2(os_ptr op)
  132. {    float wbox[10];
  133.     gs_show_enum *penum = op_show_find();
  134.     int code = num_params(op, 10, wbox);
  135.     if ( penum == 0 )
  136.         return_error(e_undefined);
  137.     if ( code < 0 ) return code;
  138.     if ( (code = gs_setcachedevice2(penum, igs, wbox[0], wbox[1], wbox[2], wbox[3], wbox[4], wbox[5], wbox[6], wbox[7], wbox[8], wbox[9])) < 0 )
  139.         return code;
  140.     pop(10);
  141.     return 0;
  142. }
  143.  
  144. /* <string> <numarray|numstring> xshow - */
  145. private int
  146. zxshow(os_ptr op)
  147. {    return moveshow(op, 1);
  148. }
  149.  
  150. /* <string> <numarray|numstring> yshow - */
  151. private int
  152. zyshow(os_ptr op)
  153. {    return moveshow(op, 2);
  154. }
  155.  
  156. /* <string> <numarray|numstring> xyshow - */
  157. private int
  158. zxyshow(os_ptr op)
  159. {    return moveshow(op, 3);
  160. }
  161.  
  162. /* Common code for {x,y,xy}show */
  163. private int
  164. moveshow(os_ptr op, int xymask)
  165. {    int code = op_show_setup(op - 1, 1);
  166.     gs_show_enum *penum = op_show_senum();
  167.     ref *psslot = op_show_psslot();
  168.     stream *s;
  169.     if ( code < 0 ) return code;
  170.     if ( (code = gs_xyshow_n_init(penum, igs, (char *)op[-1].value.bytes, r_size(op - 1)) < 0) )
  171.     {    op_show_free();
  172.         return code;
  173.     }
  174.     s = (stream *)alloc(1, sizeof(stream), "moveshow(stream)");
  175.     if ( s == 0 )
  176.         return_error(e_VMerror);
  177.     make_tasv(psslot, t_string, 0, sizeof(stream), bytes, (byte *)s);
  178.     code = sread_num_array(s, op);
  179.     if ( code < 0 )
  180.     {    op_show_free();
  181.         return code;
  182.     }
  183.     pop(2);  op -= 2;
  184.     return moveshow_continue(op, xymask);
  185. }
  186.  
  187. /* Continuation procedures */
  188.  
  189. private int
  190. xshow_continue(os_ptr op)
  191. {    return moveshow_continue(op, 1);
  192. }
  193.  
  194. private int
  195. yshow_continue(os_ptr op)
  196. {    return moveshow_continue(op, 2);
  197. }
  198.  
  199. private int
  200. xyshow_continue(os_ptr op)
  201. {    return moveshow_continue(op, 3);
  202. }
  203.  
  204. /* Get one value from the encoded number string or array. */
  205. /* Sets pvalue->value.realval. */
  206. private int
  207. sget_real(stream *s, ref *pvalue, int read)
  208. {    if ( read )
  209.     {    int code;
  210.         switch ( code = sget_encoded_number(s, pvalue) )
  211.         {
  212.         case t_integer: pvalue->value.realval = pvalue->value.intval;
  213.         case t_real: break;
  214.         case t_null: code = e_rangecheck;
  215.         default: return code;
  216.         }
  217.     }
  218.     else
  219.         pvalue->value.realval = 0;
  220.     return 0;
  221. }
  222.  
  223. private int
  224. moveshow_continue(os_ptr op, int xymask)
  225. {    int code;
  226.     stream *s = (stream *)(op_show_psslot()->value.bytes);
  227.     gs_show_enum *penum = op_show_senum();
  228. next:    code = gs_show_next(penum);
  229.     if ( code != gs_show_move )
  230.     {    code = op_show_continue_dispatch(op, code);
  231.         if ( code == o_push_estack )    /* must be gs_show_render */
  232.         {    make_op_estack(esp - 1, xyshow_continues[xymask]);
  233.         }
  234.         else if ( code < 0 )
  235.             goto errx;
  236.         return code;
  237.     }
  238.     {    /* Move according to the next value(s) from the stream. */
  239.         ref rwx, rwy;
  240.         code = sget_real(s, &rwx, xymask & 1);
  241.         if ( code < 0 ) goto errx;
  242.         code = sget_real(s, &rwy, xymask & 2);
  243.         if ( code < 0 ) goto errx;
  244.         code = gs_rmoveto(igs, rwx.value.realval, rwy.value.realval);
  245.         if ( code < 0 ) goto errx;
  246.     }
  247.     goto next;
  248. errx:    op_show_free();
  249.     return code;
  250. }
  251.  
  252. /* ------ Initialization procedure ------ */
  253.  
  254. op_def zchar2_op_defs[] = {
  255.     {"2cshow", zcshow},
  256.     {"1glyphshow", zglyphshow},
  257.     {"0rootfont", zrootfont},
  258.     {":setcachedevice2", zsetcachedevice2},
  259.     {"2xshow", zxshow},
  260.     {"2xyshow", zxyshow},
  261.     {"2yshow", zyshow},
  262.         /* Internal operators */
  263.     {"0%cshow_continue", cshow_continue},
  264.     {"0%xshow_continue", xshow_continue},
  265.     {"0%yshow_continue", yshow_continue},
  266.     {"0%xyshow_continue", xyshow_continue},
  267.     op_def_end(0)
  268. };
  269.